A blog on statistics, methods, philosophy of science, and open science. Understanding 20% of statistics will improve 80% of your inferences.

Thursday, March 12, 2020

What’s a family in family-wise error control?

When you perform multiple comparisons in a study, you need to control your alpha level for multiple comparisons. It is generally recommended to control for the family-wise error rate, but there is some confusion about what a 'family' is. As Bretz, Hothorn, & Westfall (2011) write in their excellent book “Multiple Comparisons Using R” on page 15: “The appropriate choice of null hypotheses being of primary interest is a controversial question. That is, it is not always clear which set of hypotheses should constitute the family H1,…,Hm. This topic has often been in dispute and there is no general consensus.” In one of the best papers on controlling for multiple comparisons out there, Bender & Lange (2001) write: “Unfortunately, there is no simple and unique answer to when it is appropriate to control which error rate. Different persons may have different but nevertheless reasonable opinions. In addition to the problem of deciding which error rate should be under control, it has to be defined first which tests of a study belong to one experiment.” The Wikipedia page on family-wise error rate is a mess.

I will be honest: I have never understood this confusion about what a family of tests is when controlling the family-wise error rate. At least not in a Neyman-Pearson approach to hypothesis testing, where the goal is to use data to make decisions about how to act. Neyman (Neyman, 1957) calls his approach inductive behavior. The outcome of an experiment leads one to take different possible actions, which can be either practical (e.g., implement a new procedure, abandon a research line) or scientific (e.g., claim there is or is no effect). From an error-statistical approach (Mayo, 2018) inflated Type 1 error rates mean that it has become very likely that you will be able to claim support for your hypothesis, even when the hypothesis is wrong. This reduces the severity of the test. To prevent this, we need to control our error rate at the level of our claim.

One reason the issue of family-wise error rates might remain vague, is that researchers are often vague about their claims. We do not specify our hypotheses unambiguously, and therefore this issue remains unclear. To be honest, I suspect another reason there is a continuing debate about whether and how to lower the alpha level to control for multiple comparisons in some disciplines is that 1) there are a surprisingly large number of papers written on this topic that argue you do not need to control for multiple comparisons, which are 2) cited a huge number of times giving rise to the feeling that surely they must have a point. Regrettably, the main reason these papers are written is because there are people who don't think a Neyman-Pearson approach to hypothesis testing is a good idea, and the main reason these papers are cited is because doing so is convenient for researchers who want to publish statistically significant results, as they can justify why they are not lowering their alpha level, making that p = 0.02 in one of three tests really 'significant'. All papers that argue against the need to control for multiple comparisons when testing hypotheses are wrong.  Yes, their existence and massive citation counts frustrate me. It is fine not to test a hypothesis, but when you do, and you make a claim based on a test, you need to control your error rates. 

But let’s get back to our first problem, which we can solve by making the claims people need to control Type 1 error rates for less vague. Lisa DeBruine and I recently proposed machine readable hypothesis tests to remove any ambiguity in the tests we will perform to examine statistical predictions, and when we will consider a claim corroborated or falsified. In this post, I am going to use our R package ‘scienceverse’ to clarify what constitutes a family of tests when controlling the family-wise error rate.

An example of formalizing family-wise error control


Let’s assume we collect data from 100 participants in a control and treatment condition. We collect 3 dependent variables (dv1, dv2, and dv3). In the population there is no difference between groups on any of these three variables (the true effect size is 0). We will analyze the three dv’s in independent t-tests. This requires specifying our alpha level, and thus deciding whether we need to correct for multiple comparisons. How we control error rates depends on claim we want to make.

We might want to act as if (or claim that) our treatment works if there is a difference between the treatment and control conditions on any of the three variables. In scienceverse terms, this means we consider the prediction corroborated when the p-value of the first t-test is smaller than alpha level, the p-value of the second t-test is smaller than the alpha level, or the p-value of the first t-test is smaller than the alpha level. In the scienceverse code, we specify a criterion for each test (a p-value smaller than the alpha level, p.value < alpha_level) and conclude the hypothesis is corroborated if either of these criteria are met ("p_t_1 | p_t_2 | p_t_3").  

We could also want to make three different predictions. Instead of one hypothesis (“something will happen”) we have three different hypotheses, and predict there will be an effect on dv1, dv2, and dv3. The criterion for each t-test is the same, but we now have three hypotheses to evaluate (H1, H2, and H3). Each of these claims can be corroborated, or not.

Scienceverse allows you to specify your hypotheses tests unambiguously (for code used in this blog, see the bottom of the post). It also allows you to simulate a dataset, which we can use to examine Type 1 errors by simulating data where no true effects exist. Finally, scienceverse allows you to run the pre-specified analyses on the (simulated) data, and will automatically create a report that summarizes which hypotheses were corroborated (which is useful when checking if the conclusions in a manuscript indeed follow from the preregistered analyses, or not). The output a single simulated dataset for the scenario where we will interpret any effect on the three dv’s as support for the hypothesis looks like this:

Evaluation of Statistical Hypotheses

12 March, 2020

Simulating Null Effects Postregistration

Results

Hypothesis 1: H1

Something will happen
  • p_t_1 is confirmed if analysis ttest_1 yields p.value<0.05

    The result was p.value = 0.452 (FALSE)

  • p_t_2 is confirmed if analysis ttest_2 yields p.value<0.05

    The result was p.value = 0.21 (FALSE)

  • p_t_3 is confirmed if analysis ttest_3 yields p.value<0.05

    The result was p.value = 0.02 (TRUE)

Corroboration ( TRUE )

The hypothesis is corroborated if anything is significant.
 p_t_1 | p_t_2 | p_t_3 

Falsification ( FALSE )

The hypothesis is falsified if nothing is significant.
 !p_t_1 & !p_t_2 & !p_t_3 
All criteria were met for corroboration.



We see the hypothesis that ‘something will happen’ is corroborated, because there was a significant difference on dv3 - even though this was a Type 1 error, since we simulated data with a true effect size of 0 - and any difference was taken as support for the prediction. With a 5% alpha level, we will observe 1-(1-0.05)^3 = 14.26% Type 1 errors in the long run. This Type 1 error inflation can be prevented by lowering the alpha level, for example by a Bonferroni correction (0.05/3), after which the expected Type 1 error rate is 4.92% (see Bretz et al., 2011, for more advanced techniques to control error rates). When we examine the report for the second scenario, where each dv tests a unique hypothesis, we get the following output from scienceverse:


Evaluation of Statistical Hypotheses

12 March, 2020

Simulating Null Effects Postregistration

Results

Hypothesis 1: H1

dv1 will show an effect
  • p_t_1 is confirmed if analysis ttest_1 yields p.value<0.05

    The result was p.value = 0.452 (FALSE)

Corroboration ( FALSE )

The hypothesis is corroborated if dv1 is significant.
 p_t_1 

Falsification ( TRUE )

The hypothesis is falsified if dv1 is not significant.
 !p_t_1 
All criteria were met for falsification.

Hypothesis 2: H2

dv2 will show an effect
  • p_t_2 is confirmed if analysis ttest_2 yields p.value<0.05

    The result was p.value = 0.21 (FALSE)

Corroboration ( FALSE )

The hypothesis is corroborated if dv2 is significant.
 p_t_2 

Falsification ( TRUE )

The hypothesis is falsified if dv2 is not significant.
 !p_t_2 
All criteria were met for falsification.

Hypothesis 3: H3

dv3 will show an effect
  • p_t_3 is confirmed if analysis ttest_3 yields p.value<0.05

    The result was p.value = 0.02 (TRUE)

Corroboration ( TRUE )

The hypothesis is corroborated if dv3 is significant.
 p_t_3 

Falsification ( FALSE )

The hypothesis is falsified if dv3 is not significant.
 !p_t_3 
All criteria were met for corroboration.


We now see that two hypotheses were falsified (yes, yes, I know you should not use p > 0.05 to falsify a prediction in real life, and this part of the example is formally wrong so I don't also have to explain equivalence testing to readers not familiar with it - if that is you, read this, and know scienceverse will allow you to specify equivalence test as the criterion to falsify a prediction, see the example here). The third hypothesis is corroborated, even though, as above, this is a Type 1 error.

It might seem that the second approach, specifying each dv as it’s own hypothesis, is the way to go if you do not want to lower the alpha level to control for multiple comparisons. But take a look at the report of the study you have performed. You have made 3 predictions, of which 1 was corroborated. That is not an impressive success rate. Sure, mixed results happen, and you should interpret results not just based on the p-value (but on the strength of the experimental design, assumptions about power, your prior, the strength of the theory, etc.) but if these predictions were derived from the same theory, this set of results is not particularly impressive. Since researchers can never selectively report only those results that ‘work’ because this would be a violation of the code of research integrity, we should always be able to see the meager track record of predictions.If you don't feel ready to make a specific predictions (and run the risk of sullying your track record) either do unplanned exploratory tests, and do not make claims based on their results, or preregister all possible tests you can think of, and massively lower your alpha level to control error rates (for example, genome-wide association studies sometimes use an alpha level of 5 x 10–8 to control the Type 1 erorr rate).

Hopefully, specifying our hypotheses (and what would corroborate them) transparently by using scienceverse makes it clear what happens in the long run in both scenarios. In the long run, both the first scenario, if we would use an alpha level of 0.05/3 instead of 0.05, and the second scenario, with an alpha level of 0.05 for each individual hypothesis, will lead to the same end result: Not more than 5% of our claims will be wrong, if the null hypothesis is true. In the first scenario, we are making one claim in an experiment, and in the second we make three. In the second scenario we will end up with more false claims in an absolute sense, but the relative number of false claims is the same in both scenarios. And that’s exactly the goal of family-wise error control.

References
Bender, R., & Lange, S. (2001). Adjusting for multiple testing—When and how? Journal of Clinical Epidemiology, 54(4), 343–349.
Bretz, F., Hothorn, T., & Westfall, P. H. (2011). Multiple comparisons using R. CRC Press.
Mayo, D. G. (2018). Statistical inference as severe testing: How to get beyond the statistics wars. Cambridge University Press.
Neyman, J. (1957). “Inductive Behavior” as a Basic Concept of Philosophy of Science. Revue de l’Institut International de Statistique / Review of the International Statistical Institute, 25(1/3), 7. https://doi.org/10.2307/1401671

Thanks to Lisa DeBruine for feedback on an earlier draft of this blog post.

# Scienceverse Sim
# install scienceverse
# devtools::install_github("scienceverse/scienceverse")
library(scienceverse)
library(faux)
set.seed(2) # set.seed(2) is a random draw where H1 is corroborated.
nsim <- 1
alpha_level <- 0.05
res_sim_1 <-numeric(nsim) #set up empty container for all results
for (i in 1:nsim) {
#Set up the study
sim_study <- study("Simulating Null Effects",
author = c("Daniel Lakens", "Lisa DeBruine"))
#Add a hypothesis
sim_study <- add_hypothesis(
study = sim_study,
description = "Something will happen",
id = "H1"
)
# Add an independent t-test as analysis
# Note we know our dataframe is called dat, and has a column condition, with values control & treatment
# We also have columns dv1 dv2 dv3 which are our dvs.
sim_study <- add_analysis(sim_study,
id = "ttest_1",
code = t.test(dat[which(dat$condition == "control"),]$dv1,
dat[which(dat$condition == "treatment"),]$dv1,
paired = FALSE,
conf.level = (1-alpha_level)),
software = R.version.string)
sim_study <- add_analysis(sim_study,
id = "ttest_2",
code = t.test(dat[which(dat$condition == "control"),]$dv2,
dat[which(dat$condition == "treatment"),]$dv2,
paired = FALSE,
conf.level = (1-alpha_level)),
software = R.version.string)
sim_study <- add_analysis(sim_study,
id = "ttest_3",
code = t.test(dat[which(dat$condition == "control"),]$dv3,
dat[which(dat$condition == "treatment"),]$dv3,
paired = FALSE,
conf.level = (1-alpha_level)),
software = R.version.string)
# Add criterion
sim_study <- add_criterion(
sim_study,
id = "p_t_1",
hypothesis_id = "H1",
analysis_id = "ttest_1",
result = "p.value",
operator = "<",
comparator = alpha_level)
sim_study <- add_criterion(
sim_study,
id = "p_t_2",
hypothesis_id = "H1",
analysis_id = "ttest_2",
result = "p.value",
operator = "<",
comparator = alpha_level)
sim_study <- add_criterion(
sim_study,
id = "p_t_3",
hypothesis_id = "H1",
analysis_id = "ttest_3",
result = "p.value",
operator = "<",
comparator = alpha_level)
# Add evaluation.
sim_study <- add_eval(sim_study,
hypothesis_id = "H1",
"corroboration",
description = "The hypothesis is corroborated if anything is significant.",
evaluation = "p_t_1 | p_t_2 | p_t_3")
sim_study <- add_eval(sim_study,
hypothesis_id = "H1",
"falsification",
description = "The hypothesis is falsified if nothing is significant.",
evaluation = "!p_t_1 & !p_t_2 & !p_t_3")
# Simulate some data and add it to the study
sim_study <- add_sim_data(
sim_study,
data_id = "dat",
within = list(dv = c("dv1", "dv2", "dv3")),
between = list(condition = c("control", "treatment")),
n = 100,
mu = c(100, 100, 100, 100, 100, 100),
sd = 10)
# Take a look at the data
# sim_study$data[[1]]$data
# Analyze the results
sim_study <- study_analyze(sim_study)
res_sim_1[i] <- sim_study[["hypotheses"]][[1]][["conclusion"]] == "corroborate"
if(nsim == 1){study_report(sim_study, template = "postreg", filename = "study_1.html")}
}
sum(res_sim_1/nsim)

# Scienceverse Sim
# install scienceverse
# devtools::install_github("scienceverse/scienceverse")
library(scienceverse)
library(faux)
set.seed(2) # set.seed(2) is a random draw where H1 is corroborated.
nsim <- 1
res_sim_1 <-numeric(nsim) #set up empty container for all results
res_sim_2 <-numeric(nsim) #set up empty container for all results
res_sim_3 <-numeric(nsim) #set up empty container for all results
for (i in 1:nsim) {
#Set up the study
sim_study <- study("Simulating Null Effects",
author = c("Daniel Lakens", "Lisa DeBruine"))
#Add a hypothesis
sim_study <- add_hypothesis(
study = sim_study,
description = "dv1 will show an effect",
id = "H1"
)
sim_study <- add_hypothesis(
study = sim_study,
description = "dv2 will show an effect",
id = "H2"
)
sim_study <- add_hypothesis(
study = sim_study,
description = "dv3 will show an effect",
id = "H3"
)
# Add an independent t-test as analysis
# Note we know our dataframe is called dat, and has a column condition, with values control & treatment
# We also have columns dv1 dv2 dv3 which are our dvs.
sim_study <- add_analysis(sim_study,
id = "ttest_1",
code = t.test(dat[which(dat$condition == "control"),]$dv1,
dat[which(dat$condition == "treatment"),]$dv1,
paired = FALSE,
conf.level = (1-alpha_level)),
software = R.version.string)
sim_study <- add_analysis(sim_study,
id = "ttest_2",
code = t.test(dat[which(dat$condition == "control"),]$dv2,
dat[which(dat$condition == "treatment"),]$dv2,
paired = FALSE,
conf.level = (1-alpha_level)),
software = R.version.string)
sim_study <- add_analysis(sim_study,
id = "ttest_3",
code = t.test(dat[which(dat$condition == "control"),]$dv3,
dat[which(dat$condition == "treatment"),]$dv3,
paired = FALSE,
conf.level = (1-alpha_level)),
software = R.version.string)
# Add criterion
sim_study <- add_criterion(
sim_study,
id = "p_t_1",
hypothesis_id = "H1",
analysis_id = "ttest_1",
result = "p.value",
operator = "<",
comparator = alpha_level)
sim_study <- add_criterion(
sim_study,
id = "p_t_2",
hypothesis_id = "H2",
analysis_id = "ttest_2",
result = "p.value",
operator = "<",
comparator = alpha_level)
sim_study <- add_criterion(
sim_study,
id = "p_t_3",
hypothesis_id = "H3",
analysis_id = "ttest_3",
result = "p.value",
operator = "<",
comparator = alpha_level)
# Add evaluation.
sim_study <- add_eval(sim_study,
hypothesis_id = "H1",
"corroboration",
description = "The hypothesis is corroborated if dv1 is significant.",
evaluation = "p_t_1")
sim_study <- add_eval(sim_study,
hypothesis_id = "H2",
"corroboration",
description = "The hypothesis is corroborated if dv2 is significant.",
evaluation = "p_t_2")
sim_study <- add_eval(sim_study,
hypothesis_id = "H3",
"corroboration",
description = "The hypothesis is corroborated if dv3 is significant.",
evaluation = "p_t_3")
sim_study <- add_eval(sim_study,
hypothesis_id = "H1",
"falsification",
description = "The hypothesis is falsified if dv1 is not significant.",
evaluation = "!p_t_1")
sim_study <- add_eval(sim_study,
hypothesis_id = "H2",
"falsification",
description = "The hypothesis is falsified if dv2 is not significant.",
evaluation = "!p_t_2")
sim_study <- add_eval(sim_study,
hypothesis_id = "H3",
"falsification",
description = "The hypothesis is falsified if dv3 is not significant.",
evaluation = "!p_t_3")
# Simulate some data and add it to the study
sim_study <- add_sim_data(
sim_study,
data_id = "dat",
within = list(dv = c("dv1", "dv2", "dv3")),
between = list(condition = c("control", "treatment")),
n = 100,
mu = c(100, 100, 100, 100, 100, 100),
sd = 10)
# Take a look at the data
# sim_study$data[[1]]$data
# Analyze the results
sim_study <- study_analyze(sim_study)
res_sim_1[i] <- sim_study[["hypotheses"]][[1]][["conclusion"]] == "corroborate"
res_sim_2[i] <- sim_study[["hypotheses"]][[2]][["conclusion"]] == "corroborate"
res_sim_3[i] <- sim_study[["hypotheses"]][[3]][["conclusion"]] == "corroborate"
if(nsim == 1){study_report(sim_study, template = "postreg", filename = "study_2.html")}
}
sum(res_sim_1/nsim)
sum(res_sim_2/nsim)
sum(res_sim_3/nsim)

1 comment:

  1. This comment has been removed by a blog administrator.

    ReplyDelete